using System;
using System.Collections;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;

namespace FreeFlow.Administration
{
	/// <summary>
	/// A collection of actions.
	/// </summary>
	public sealed class ActionCollection : DatabaseAware, IRefreshable, IList, IBindingList, IEnumerable<Action>
	{
    private Map map;
    private string stageName;

    /// <summary>
    /// Initializes a new instance of the <see cref="ActionCollection"/> class for the specified stage in the specified map.
    /// </summary>
    /// <param name="server">The server.</param>
    /// <param name="mapName">Name of the map.</param>
    /// <param name="stageName">Name of the stage.</param>
		public ActionCollection(Server server, string mapName, string stageName) : base(server)
		{
      this.map = Server.Maps[mapName];
      this.stageName = stageName;
		}

    /// <summary>
    /// Initializes a new instance of the <see cref="ActionCollection"/> class for the specified map. Will create a collection of creation actions.
    /// </summary>
    /// <param name="server">The server.</param>
    /// <param name="mapName">Name of the map.</param>
    public ActionCollection(Server server, string mapName) : this(server, mapName, string.Empty)
    {
    }

    private List<Action> actions;
    private void GetActions()
    {
      if (actions == null)
      {
        actions = new List<Action>();
        string sql;
        if (string.IsNullOrEmpty(stageName))
          sql = string.Format("SELECT * FROM eAction LEFT OUTER JOIN eStart " +
            "ON eAction.eMapName = eStart.eMapName AND eAction.eActionName = eStart.eActionName " +
            "WHERE eAction.eMapName = '{0}' AND (eAction.eStageName = '' OR eAction.eStageName = ' ') ORDER BY eActionID", 
            SqlEscape(map.Name));
        else
          sql = string.Format("SELECT * FROM eAction WHERE eMapName = '{0}' AND eStageName = '{1}' ORDER BY eActionID", 
            SqlEscape(map.Name), SqlEscape(stageName));

        // get actions
        using (IDataReader reader = Server.ExecuteReader(sql))
        {
          while (reader.Read()) 
          {
            Action action = new Action(Server, map, reader);
            actions.Add(action);
          }
        }
      }
    }

    IEnumerator IEnumerable.GetEnumerator()
    {
      GetActions();
      return actions.GetEnumerator();
    }

    #region IEnumerable<Action> Members

    /// <summary>
    /// Returns an enumerator that iterates through the collection.
    /// </summary>
    /// <returns>
    /// A <see cref="T:System.Collections.Generic.IEnumerator`1"/> that can be used to iterate through the collection.
    /// </returns>
    public IEnumerator<Action> GetEnumerator()
    {
      GetActions();
      return actions.GetEnumerator();
    }

    #endregion

    /// <summary>
    /// Gets the number of actions in this collection.
    /// </summary>
    public int Count
    {
      get
      {
        GetActions();
        return actions.Count;
      }
    }

    /// <summary>
    /// Gets the <see cref="Action"/> at the specified index.
    /// </summary>
    public Action this[int index]
    {
      get
      {
        GetActions();
        return (Action)actions[index];
      }
    }

    /// <summary>
    /// Gets the <see cref="Action"/> with the specified name.
    /// </summary>
    public Action this[string name]
    {
      get
      {
        GetActions();
        for (int i = 0; i < Count; i++)
        {
          if (this[i].Name == name)
            return this[i];
        }

        return null;
      }
    }

    #region IRefreshable Members

    /// <summary>
    /// Refreshes the list of actions.
    /// </summary>
    public void Refresh()
    {
      actions = null;
    }

    #endregion
    #region IList Members

    /// <summary>
    /// Gets a value indicating whether the <see cref="T:System.Collections.IList"/> is read-only.
    /// </summary>
    public bool IsReadOnly
    {
      get
      {
        return true;
      }
    }

    object IList.this[int index]
    {
      get
      {
        GetActions();
        return actions[index];
      }
      set
      {
        throw new NotSupportedException(readOnlyListMessage);
      }
    }

    void IList.RemoveAt(int index)
    {
      throw new NotSupportedException(readOnlyListMessage);
    }

    void IList.Insert(int index, object value)
    {
      throw new NotSupportedException(readOnlyListMessage);
    }

    void IList.Remove(object value)
    {
      throw new NotSupportedException(readOnlyListMessage);
    }

    /// <summary>
    /// Determines whether the <see cref="T:System.Collections.IList"/> contains a specific value.
    /// </summary>
    /// <param name="value">The <see cref="T:System.Object"/> to locate in the <see cref="T:System.Collections.IList"/>.</param>
    /// <returns>
    /// 	<see langword="true"/> if the <see cref="T:System.Object"/>
    /// is found in the <see cref="T:System.Collections.IList"/>; otherwise, <see langword="false"/>.
    /// </returns>
    public bool Contains(object value)
    {
      GetActions();
      return actions.Contains((Action)value);
    }

    void IList.Clear()
    {
      throw new NotSupportedException(readOnlyListMessage);
    }

    /// <summary>
    /// Determines the index of a
    /// specific item in the <see cref="T:System.Collections.IList"/>.
    /// </summary>
    /// <param name="value">The <see cref="T:System.Object"/> to locate in the <see cref="T:System.Collections.IList"/>.</param>
    /// <returns>
    /// The index of <paramref name="value"/> if found in the list; otherwise, -1.
    /// </returns>
    public int IndexOf(object value)
    {
      GetActions();
      return actions.IndexOf((Action)value);
    }

    int IList.Add(object value)
    {
      throw new NotSupportedException(readOnlyListMessage);
    }

    /// <summary>
    /// Gets a value indicating whether the <see cref="T:System.Collections.IList"/> has a fixed size.
    /// </summary>
    public bool IsFixedSize
    {
      get
      {
        return true;
      }
    }

    #endregion
    #region ICollection Members

    /// <summary>
    /// Gets a value
    /// indicating whether access to the <see cref="T:System.Collections.ICollection"/> is synchronized
    /// (thread-safe).
    /// </summary>
    public bool IsSynchronized
    {
      get
      {
        return false;
      }
    }

    /// <summary>
    /// Copies the elements of
    /// the <see cref="T:System.Collections.ICollection"/> to an <see cref="T:System.Array"/>, starting at a particular <see cref="T:System.Array"/> index.
    /// </summary>
    /// <param name="array">The one-dimensional <see cref="T:System.Array"/> that is the destination of the elements copied from <see cref="T:System.Collections.ICollection"/>. The <see cref="T:System.Array"/> must have zero-based indexing.</param>
    /// <param name="index">The zero-based index in <paramref name="array"/> at which copying begins.</param>
    /// <exception cref="T:System.ArgumentNullException">
    /// 	<paramref name="array"/> is <see langword="null"/>.</exception>
    /// <exception cref="T:System.ArgumentOutOfRangeException">
    /// 	<paramref name="index"/> is less than zero.</exception>
    /// <exception cref="T:System.ArgumentException">
    /// 	<para>
    /// 		<paramref name="array"/> is multidimensional.</para>
    /// 	<para>-or-</para>
    /// 	<para>
    /// 		<paramref name="index"/> is equal to or greater than the length of <paramref name="array"/>.</para>
    /// 	<para>-or-</para>
    /// 	<para>The number of elements in the source <see cref="T:System.Collections.ICollection"/> is greater than the available space from <paramref name="index"/> to the end of the destination <paramref name="array"/>.</para>
    /// </exception>
    /// <exception cref="T:System.InvalidCastException">The type of the source <see cref="T:System.Collections.ICollection"/> cannot be cast automatically to the type of the destination <paramref name="array"/>.</exception>
    public void CopyTo(Action[] array, int index)
    {
      GetActions();
      actions.CopyTo(array, index);
    }

    void ICollection.CopyTo(Array array, int index)
    {
      CopyTo((Action[])array, index);
    }

    /// <summary>
    /// Gets an object that
    /// can be used to synchronize access to the <see cref="T:System.Collections.ICollection"/>.
    /// </summary>
    public object SyncRoot
    {
      get
      {
        return null;
      }
    }

    #endregion
    #region IBindingList Members

    void IBindingList.ApplySort(PropertyDescriptor property, ListSortDirection direction)
    {
      GetActions();
      sortProperty = property;
      sortDirection = direction;
      actions.Sort(new Comparer<Action>(sortProperty, sortDirection));
    }

    private PropertyDescriptor sortProperty;
    PropertyDescriptor IBindingList.SortProperty
    {
      get
      {
        return sortProperty;
      }
    }

    bool IBindingList.SupportsSorting
    {
      get
      {
        return true;
      }
    }

    bool IBindingList.IsSorted
    {
      get
      {
        return (sortProperty != null);
      }
    }

    void IBindingList.RemoveSort()
    {
      sortProperty = null;
    }

    private ListSortDirection sortDirection;
    ListSortDirection IBindingList.SortDirection
    {
      get
      {
        return sortDirection;
      }
    }

    #region not implemented 
    int IBindingList.Find(PropertyDescriptor property, object key)
    {
      return 0;
    }

    bool IBindingList.AllowRemove
    {
      get
      {
        return false;
      }
    }

    bool IBindingList.SupportsSearching
    {
      get
      {
        return false;
      }
    }

    private ListChangedEventHandler listChanged;
    event ListChangedEventHandler IBindingList.ListChanged
    {
      add { listChanged += value; }
      remove { listChanged -= value; }

    }

    bool IBindingList.SupportsChangeNotification
    {
      get
      {
        return false;
      }
    }

    object IBindingList.AddNew()
    {
      return null;
    }

    bool IBindingList.AllowEdit
    {
      get
      {
        return false;
      }
    }

    void IBindingList.RemoveIndex(PropertyDescriptor property)
    {
    }

    void IBindingList.AddIndex(PropertyDescriptor property)
    {
    }

    bool IBindingList.AllowNew
    {
      get
      {
        return false;
      }
    }

    #endregion
    #endregion
  }
}
